### **Lab 8: LCD Display and Interrupts**

### Submit hello blink.asm by 5:00pm on March 16, 2018.

## I. LCD Display

Download LCDdefs.inc, lcd\_function\_defs.inc, lcd\_function\_code.asm and hello\_world.asm. The HD44780 LCD Driver for ATmega2560 is written in LCDdefs.inc , lcd\_function\_defs.inc and lcd\_function\_code.asm. The file "hello\_world.asm" (by Dr. Mike Zastre) shows how to display strings on LCD.

|       | col 0 |   |   |   |   |   |  |  |  |  | col 15 |
|-------|-------|---|---|---|---|---|--|--|--|--|--------|
| row 0 | С     | S | С | 2 | 3 | 0 |  |  |  |  |        |
| row 1 |       |   |   |   |   |   |  |  |  |  |        |

To display a character on the LCD, step1 - initialize the LCD according to the specification in the Hitachi HD44780 datasheet; step2 - set the row, column numbers correctly, for example, in the diagram above, the first 'C' is displayed at row 0, column 0; step3 - display the character. The following code shows how to display a character 'U' at row 1, column 7:

```
.cseg
.org O
                                   - More on interrupt later in this lab.
    jmp start
.include "m2560def.inc"
                                                  Must include the driver files.
.include "lcd_function_defs.inc"
_include "lcd_function_code.asm"
.cseg
start:
                                   Initialize the LCD according to the specification
    rcall lcd_init
                                   in the Hitachi HD44780 datasheet.
                          lcd_gotoxy(row, col)
     ldi r16, 1
                   :rew
     ldi r17, 4
                   ;column
    push r16
                                       Set the cursor at (row, column)
    push r17
    rcall lcd gotoxy
     pop r17
     pop_r16
               'U'
                     ;lcd_putchar(char)
     ídi r16,
    push r16
    rcall lcd_putchar
    pop r16
                                           Display 'U' at (1, 4) row 1, column 4
Done: rjmp Done
```

II. Exercises: in hello\_world.asm, build the file and observe the output on LCD display, then change the code so that it displays "Hello, world!" starting at row 1, column 3.

# III. Timer/Counter1 Interrupt

An interrupt is a signal that can interrupt and alter the flow of current program execution. It is a method that allows programs to respond immediately to external events. Interrupts can be triggered by external or internal signals; they can be caused by software or hardware. When a program is interrupted, the routine that is executed in response to the interrupt is called an **interrupt service routine** (**ISR**), or **interrupt handler**. The process of interrupting the current program, executing the interrupt handler, and returning, is called **servicing the interrupt**. Interrupts can be dedicated or shared. If interrupts are shared among multiple devices (I/O), then the ISR further has to check which device or signal caused that interrupt. Usually the interrupts are numbered and when an interrupt occurs, the processor jumps to a pre-defined location assigned to that interrupt. Hence an interrupt vector table has to be setup to jump and branch to appropriate ISR to handle interrupts. The ATMega 2560 processor we use in the lab has following interrupt vector table (p101 of <sup>1</sup>):

Table 13-1. Reset and Interrupt Vectors

| Vector Program No. Address <sup>(2)</sup> |                       | Source       | Interrupt Definition                                                                |  |  |  |  |  |
|-------------------------------------------|-----------------------|--------------|-------------------------------------------------------------------------------------|--|--|--|--|--|
| 1                                         | \$0000(1)             | RESET        | External Pin, Power-on Reset, Brown-out Reset<br>Watchdog Reset, and JTAG AVR Reset |  |  |  |  |  |
| 2                                         | \$0002                | INT0         | External Interrupt Request 0                                                        |  |  |  |  |  |
| 3                                         | \$0004                | INT1         | External Interrupt Request 1                                                        |  |  |  |  |  |
| 4                                         | \$0006                | INT2         | External Interrupt Request 2                                                        |  |  |  |  |  |
| 5                                         | \$0008                | INT3         | External Interrupt Request 3                                                        |  |  |  |  |  |
| 6                                         | \$000A                | INT4         | External Interrupt Request 4                                                        |  |  |  |  |  |
| 7                                         | \$000C                | INT5         | External Interrupt Request 5                                                        |  |  |  |  |  |
| 8                                         | \$000E                | INT6         | External Interrupt Request 6                                                        |  |  |  |  |  |
| 9                                         | \$0010                | INT7         | External Interrupt Request 7                                                        |  |  |  |  |  |
| 10                                        | \$0012                | PCINT0       | Pin Change Interrupt Request 0                                                      |  |  |  |  |  |
| 11                                        | \$0014                | PCINT1       | Pin Change Interrupt Request 1                                                      |  |  |  |  |  |
| 12                                        | \$0016 <sup>(3)</sup> | PCINT2       | Pin Change Interrupt Request 2                                                      |  |  |  |  |  |
| 13                                        | \$0018                | WDT          | Watchdog Time-out Interrupt                                                         |  |  |  |  |  |
| 14                                        | \$001A                | TIMER2 COMPA | Timer/Counter2 Compare Match A                                                      |  |  |  |  |  |
| 15                                        | \$001C                | TIMER2 COMPB | Timer/Counter2 Compare Match B                                                      |  |  |  |  |  |
| 16                                        | \$001E                | TIMER2 OVF   | Timer/Counter2 Overflow                                                             |  |  |  |  |  |
| 17                                        | \$0020                | TIMER1 CAPT  | Timer/Counter1 Capture Event                                                        |  |  |  |  |  |
| 18                                        | \$0022                | TIMER1 COMPA | Timer/Counter1 Compare Match A                                                      |  |  |  |  |  |
| 19                                        | \$0024                | TIMER1 COMPB | Timer/Counter1 Compare Match B                                                      |  |  |  |  |  |
| 20                                        | \$0026                | TIMER1 COMPC | Timer/Counter1 Compare Match C                                                      |  |  |  |  |  |
| 21                                        | \$0028                | TIMER1 OVF   | Timer/Counter1 Overflow                                                             |  |  |  |  |  |
| 22                                        | \$002A                | TIMERO COMPA | Timer/Counter0 Compare Match A                                                      |  |  |  |  |  |
| 23                                        | \$002C                | TIMERO COMPB | Timer/Counter0 Compare match B                                                      |  |  |  |  |  |
| 24                                        | \$002E                | TIMERO OVF   | Timer/Counter0 Overflow                                                             |  |  |  |  |  |
| 25                                        | \$0030                | SPI, STC     | SPI Serial Transfer Complete                                                        |  |  |  |  |  |
| 26                                        | \$0032                | USART0 RX    | USART0 Rx Complete                                                                  |  |  |  |  |  |
| 27                                        | \$0034                | USARTO UDRE  | USART0 Data Register Empty                                                          |  |  |  |  |  |
| 28                                        | \$0036                | USART0 TX    | USART0 Tx Complete                                                                  |  |  |  |  |  |
| 29                                        | \$0038                | ANALOG COMP  | Analog Comparator                                                                   |  |  |  |  |  |

### CSC 230 Spring 2018

There are 6 built-in timers (two 8-bit and four 16-bit timer counters) in AVR processor. They can be setup in different modes. Set Timer1 in CTC (Clear Timer on Compare Match) mode. That is, the 16-bit timer is initialized to a value (say 0) and the CPU clock signal is used as source. When the timer reaches the value (called TOP) set in OCR1A (Output Compare Register), it generates an interrupt (Vector 18, see the table above) causing the processor to jump to a pre-defined location 0x0022. Let's learn the structure of interrupt in AVR assembly language:

```
.cseg
.org 0
                                         Interrupt Vector Table (IVT) for each interrupt
         ;reset
         jmp start
           ;Timer/Counter1 Compare Match A Interrupt
                                ; jump to interrupt service routine
         jmp swap_chars_isr
.include "m2560def.inc"
.include "lcd_function_defs.inc"
.include "lcd_function_code.asm"
                                        Bit 1 of TIMSK1 is the Timer1 Output Compare A
                                        Match Interrupt Enable. Set it to 1 to signal the timer
.cseg
                                        change from TOP to 0. "sei" set the I flag in SREG to 1.
start:
                                        Combine both, the Timer1 OCAM Interrupt is enabled.
                                        Then the corresponding Interrupt Vector is executed
    enable timer interrupt;
                                        when OCF1A flag (in TIFR1) is set.
    1di r16, (1 << OCIE1A)
    sts TIMSK1, r16
    ;enamble Interrupt bit,
                               /in status register
    nop
    loop:
                        The infinite loop is interrupted when timer 1
         nop
    rjmp loop
                        reaches the TOP (in OCR1A)
swap_chars_isr: ;Interrupt service routine
    push r16
    push r17
    lds r16, SREG ; reserve the content of SREG on stack
    push r16
    nop
    ;timer interrupt flag is automatically
    cleared when this ISR is executed;
    ;(p163 ATmega datasheet)
    pop r16
    sts SREG, r16
    pop r17
    pop r16
    reti
```

At power on (RESET), the program control jumps to location 0x0000. This is where you need to initialize the timer for the desired functionality. In the above program, the control jumps to start

which initializes the timer and sets in the desired mode (CTC) and then waits in an infinite loop. After the initialization, timer counter automatically starts counting with the specified clock rate and causes an interrupt when the counter reaches the TOP (OCR1A). This interrupt (18) takes the program control to location 0x0022 that further jumps to the ISR (swap chars isr).

The timer of the AVR can be specified to monitor several events. Status flags in the TIFR (Timer Interrupt Flag Register) show if an event has occurred. Some of the modes the timers can operate are: a) Timer Overflow mode; b) Compare Match and c) Input capture. For the purpose of this lab, we are going to setup the timer to operate only in Clear Timer on Compare Match (CTC) mode. The OCF1A flag is set when timer1 has counted up to the TOP value (stored in OCR1A) and is reset to zero in the next timer clock cycle. Below is the summary of the registers associated with Timer/Counter 1:

- TIMSK1 is the interrupt enable register. In order to enable the interrupt by the timer, a flag has to be set in the register as well as enable the global interrupt in SREG (using SEI instruction).
- The 16-bit timer counter value is loaded at: TCNT1H:TCNT1L. The counter will start counting up from the initial value loaded here and count up to TOP in OCR1A. In the next clock pulse, the 16-bit counter will be reset to 0x0000 and causes an interrupt.
- The timer mode (CTC mode) is set by configure to TCCR1A and TCCR1B control registers.
- To count, the timer must use a clock signal (either internal or external). In this lab, the timer is setup to use an internal clock. Note that the CPU clock is operating at 16Mhz. We will use this clock but reduce the counting speed by setting up a pre-scaler of 1024. That is the clock signal is divided by 1024 to reduce the counting speed. This is done by configuring the TCCR1B control register.

### IV. Timer/Counter 1 in AVR ATmega 2560

In this lab, the 16-bit Timer/Counter 1 is used. The full names of the registers are:

.equ TCCR1A = 0x80 Timer/Counter 1 Control Register A

.equ TCCR1B = 0x81 Timer/Counter 1 Control Register B

.equ TCCR1C = 0x82 Timer/Counter 1 Control Register C (no need to set it)

.equ TCNT1H = 0x85 high byte of the Timer/Counter 1 (no need to set it)

.equ TCNT1L = 0x84 low byte of the Timer/Counter 1 (no need to set it)

.equ TIFR1 = 0x36Timer/Counter 1 Interrupt Flag Register (no need to set it)

.equ TIMSK1 = 0x6F Timer/Counter 1 Interrupt Mask Register = 0x5F Status Register .equ SREG Timer1 Output Compare A Match Interrupt Enable, set it to 1. For the detailed description, refer to the lab 7 note. TIMSK1 - Timer/Counter 1 Interrupt Mask Register Bit 0 (0x6F) ICIE1 OCIE1C OCIE1B OCIE1A TOIE1 TIMSK1 Read/Write R R/W Initial Value 0 0 0 0 0 0

### "Bit 1 – OCIE1A: Timer/Counter1, Output Compare A Match Interrupt Enable

When this bit is written to one, and the I-flag in the Status Register is set (interrupts globally enabled), the Timer/Counter Output Compare A Match interrupt is enabled. The corresponding Interrupt Vector is executed when the OCF1A Flag (in TIFR1) is set (p162 of <sup>1</sup>). The program control jumps to address 0x0022 and further jumps to swap chars isr.

Combine the four bits in TCCR1A and TCCR1B configure the timer/counter1 mode (CTC):

TCCR1B
Table 17-2. Waveform Generation Mode Bit Description TCCR1A

| Mode | WGMn3 | WGMn2<br>(CTCn) | WGMn1<br>(PWMn1) | WGMn0<br>(PWMn0) | Timer/Counter<br>Mode of Operation  | тор    | Update of<br>OCRnX at | TOVn Flag<br>Set on |
|------|-------|-----------------|------------------|------------------|-------------------------------------|--------|-----------------------|---------------------|
| 0    | 0     | 0               | 0                | 0                | Normal                              | 0xFFFF | Immediate             | MAX                 |
| 1    | 0     | 0               | 0                | 1                | PWM, Phase Correct, 8-bit           | 0x00FF | TOP                   | воттом              |
| 2    | 0     | 0               | 1                | 0                | PWM, Phase Correct, 9-bit           | 0x01FF | TOP                   | воттом              |
| 3    | 0     | 0               | 1                | 1                | PWM, Phase Correct, 10-bit          | 0x03FF | TOP                   | воттом              |
| 4    | 0     | 1               | 0                | 0                | CTC                                 | OCRnA  | Immediate             | MAX                 |
| 5    | 0     | 1               | 0                | 1                | Fast PWM, 8-bit                     | 0x00FF | воттом                | TOP                 |
| 6    | 0     | 1               | 1                | 0                | Fast PWM, 9-bit                     | 0x01FF | воттом                | TOP                 |
| 7    | 0     | 1               | 1                | 1                | Fast PWM, 10-bit                    | 0x03FF | воттом                | TOP                 |
| 8    | 1     | 0               | 0                | 0                | PWM, Phase and Frequency<br>Correct | ICRn   | воттом                | воттом              |
| 9    | 1     | 0               | 0                | 1                | PWM,Phase and Frequency<br>Correct  | OCRnA  | воттом                | воттом              |
| 10   | 1     | 0               | 1                | 0                | PWM, Phase Correct                  | ICRn   | TOP                   | воттом              |
| 11   | 1     | 0               | 1                | 1                | PWM, Phase Correct                  | OCRnA  | TOP                   | воттом              |
| 12   | 1     | 1               | 0                | 0                | стс                                 | ICRn   | Immediate             | MAX                 |
| 13   | 1     | 1               | 0                | 1                | (Reserved)                          | -      | -                     | -                   |
| 14   | 1     | 1               | 1                | 0                | Fast PWM                            | ICRn   | воттом                | TOP                 |
| 15   | 1     | 1               | 1                | 1                | Fast PWM                            | OCRnA  | воттом                | TOP                 |

V. Exercises: Download hello\_blink.asm. Implement the interrupt service routine called swap\_chars\_isr. Display the following message and make '!' blink.



Submit hello\_blink.asm by 5:00pm on March 16, 2018.

1. Section 14/17 (p101, p133) of the ATmega2560-datasheet.pdf at <a href="http://ww1.microchip.com/downloads/en/DeviceDoc/Atmel-2549-8-bit-AVR-Microcontroller-ATmega640-1280-1281-2560-2561\_datasheet.pdf">http://ww1.microchip.com/downloads/en/DeviceDoc/Atmel-2549-8-bit-AVR-Microcontroller-ATmega640-1280-1281-2560-2561\_datasheet.pdf</a>.